import { useMemberStore } from '@/stores/index'
/**
 *添加拦截器：
 * 拦截器request请求
 * 拦截uploadfile文件上传
 * TODO:
 * 1.非http开头需拼接地址
 * 2.请求超时
 * 3.添加小程序段请求头标识
 * 4.添加token请求头标识
 */
const baseURL = 'https://pcapi-xiaotuxian-front-devtest.itheima.net'

//添加拦截器
//拦截器公共配置
const httpInterceptor = {
  //拦截前触发
  invoke(options: UniApp.RequestOptions) {
    //1.非http的路径进行拼接
    if (!options.url.startsWith('http')) {
      options.url = baseURL + options.url
    }
    //2.修改请求超时时间，默认60s太长了
    ;(options.timeout = 10000),
      //3.添加小程序端请求头标识
      (options.header = {
        ...options.header,
        'source-client': 'miniapp',
      })
    //4.添加token
    const memberStore = useMemberStore()
    const token = memberStore.profile?.token
    if (token) {
      options.header.Authorization = token
    }
  },
}
uni.addInterceptor('request', httpInterceptor)
uni.addInterceptor('uploadFile', httpInterceptor)

/**
 * 请求函数
 * @param Uniapp.RequestOptions
 * @returns Promise
 * 1.返回Promise对象
 * 2.请求成功
 *  2.1提取核心数据res.data
 *  2.2添加类型，支持泛型
 * 3.请求失败
 *  3.1网络错误->提示用户换网络
 *  3.2 401错误->清理用户信息，跳转到登录页
 *  3.3其他错误->根据后端错误信息轻提示
 */

interface Data<T> {
  code: string
  msg: string
  result: T
}
export const http = <T>(options: UniApp.RequestOptions) => {
  //1.返回Promise对象
  return new Promise<Data<T>>((resolve, reject) => {
    uni.request({
      ...options,
      //响应成功
      success(res) {
        //状态码2xx，axios就是这样设计的
        if (res.statusCode >= 200 && res.statusCode < 300) {
          //2.1 提取核心数据res.data
          resolve(res.data as Data<T>)
        } else if (res.statusCode === 401) {
          //401错误 ->清理用户信息，跳转到登录页
          const memberStore = useMemberStore()
          memberStore.clearProfile()
          uni.navigateTo({ url: '/pages/login/login' })
          reject(res)
        } else {
          //其他错误->根据后端错误信息轻提示
          uni.showToast({
            icon: 'none',
            title: (res.data as Data<T>).msg || '请求错误',
          })
          reject(res)
        }
      },
      //响应失败
      fail(err) {
        uni.showToast({
          icon: 'none',
          title: '网络错误，换个网络试试。',
        })
        reject(err)
      },
    })
  })
}
